查看原文
其他

# Pragma Pack(n)字节对齐

小师弟 果果小师弟 2022-07-15
点击上方“果果小师弟”,选择“置顶/星标公众号

干货福利,第一时间送达!

 #Pragma Pack主要是用在字节对齐方面,为什么要对齐呢?

因为计算机中内存空间都是按照byte划分的,从理论上讲似乎对任何类型的变量的访问可以从任何地址开始,但实际情况是在访问特定变量的时候经常在特定的内存地址访问,这就需要各类型数据按照一定的规则在空间上排列,而不是顺序的一个接一个的排放,这就是对齐。

#Pragma Pack (n)指定对齐的字节数

1、基本数据类型所占内存大小

#include<stdio.h>
char           a;
short          b;
int            c;
float          d;
double         e;
long           f;
unsigned char  g;
unsigned short h;
unsigned int   i;
int main()
{
printf("%d\t%d\t%d\t%d\t%d\t%d\t%d\t%d\t%d\t\n",sizeof(a),sizeof(b),sizeof(c),sizeof(d),sizeof(e),sizeof(f),sizeof(g),sizeof(h),sizeof(i));
return 0;
}

2、对齐规则

1)未指定#pragma pack时,系统默认的对齐模数4字节(32位机,X86系统等)。
2)指定#pragma pack 对齐模数时,实际取pack 对齐模数和默认的最小值。
3)结构体里面static变量,因为静态变量的存放位置与结构体实例的存储地址无关,是单独存放在静态数据区的,因此用siezof计算其大小时没有将静态成员所占的空间计算进来。

3、举例说明

默认按四个字节对齐,如果不足4个字节,就按4个字节算。如果大于4个字节,就不用管,是多少字节就是多少字节。

#include<stdio.h>
typedef struct student
{

  char           a;  //默认4个字节对齐,char           是1个字节,以4字节对齐,按4个字节处理
  short          b;  //默认4个字节对齐,short          是2个字节,以4字节对齐,按4个字节处理
  int            c;  //默认4个字节对齐,int            是4个字节,以4字节对齐,按4个字节处理
  float          d;  //默认4个字节对齐,float          是4个字节,以4字节对齐,按4个字节处理
  double         e;  //默认4个字节对齐,double         是8个字节,以4字节对齐,按8个字节处理
  long           f;  //默认4个字节对齐,long           是4个字节,以4字节对齐,按4个字节处理
  unsigned char  g;  //默认4个字节对齐,unsigned char  是1个字节,以4字节对齐,按4个字节处理
  unsigned short h;  //默认4个字节对齐,unsigned short 是2个字节,以4字节对齐,按4个字节处理
  unsigned int   i;  //默认4个字节对齐,unsigned int   是4个字节,以4字节对齐,按4个字节处理
}student;
int main()
{
  printf("%d",sizeof(student));
  return 0;
}

假如我们不使用字节对齐,那么这个结构体就是1+2+4+4+8+4+1+2+4=28个字节,但是我们实际打印却是40个字节。


这是因为编译器默认以4个字节对齐,不足4个字节(比如short和char),按4个字节处理。大于等于4个字节的就不用管。(比如double是8个字节,就按8个字节算)。
但是在实际工程中,如果我们是读取一块一块的数据,这些数据都是连在一起的,比如bmp图片,前14个字节是文件信息头,紧接着是40个字节的图像信息头。如果我们不用结构体对齐操作的话。那就乱了套了,数据就读取失败。
所以我们要在结构体前加上#Pragma Pack(1),以一个字节对齐,使用完后要加#pragma pack(),释放内存对齐。为啥要以一个字节对齐呢?是因为一般的数据类型都是大于等于1个字节的。这样的话就会按照数据类型原有的的分配。就不会错位了。

具体请看

#include<stdio.h>
#pragma pack(1)
typedef struct student
{

  char           a;  //设置1个字节对齐,char           是1个字节,以1字节对齐,按1个字节处理
  short          b;  //设置1个字节对齐,short          是2个字节,以1字节对齐,按2个字节处理
  int            c;  //设置1个字节对齐,int            是4个字节,以1字节对齐,按4个字节处理
  float          d;  //设置1个字节对齐,float          是4个字节,以1字节对齐,按4个字节处理
  double         e;  //设置1个字节对齐,double         是8个字节,以1字节对齐,按8个字节处理
  long           f;  //设置1个字节对齐,long           是4个字节,以1字节对齐,按4个字节处理
  unsigned char  g;  //设置1个字节对齐,unsigned char  是1个字节,以1字节对齐,按1个字节处理
  unsigned short h;  //设置1个字节对齐,unsigned short 是2个字节,以1字节对齐,按2个字节处理
  unsigned int   i;  //设置1个字节对齐,unsigned int   是4个字节,以1字节对齐,按4个字节处理
}student;
#pragma pack()
int main()
{
  printf("%d",sizeof(student));
  return 0;
}

这回就按照每个变量的类型给出实际的结果了,1+2+4+4+8+4+1+2+4=30


怎么样,现在知道内存对齐#pragma pack(n) 的用法了吧。

END

微信公众号:果果小师弟
关注可了解获取更多的免费资料;
如果你觉得文章对你有帮助,转发分享,欢迎赞赏

最后

10T资源大放送!包括但不限于:C/C++,Linux,Python,Java,PHP,人工智能,PCB、FPGA、DSP、单片机、等等!在公众号内回复「更多资源」,即可免费获取,期待你的关注!

您可能也对以下帖子感兴趣

文章有问题?点此查看未经处理的缓存